# Introduction to Keil IDE and MCB1700

Irene Huang

#### **KEIL IDE INTRODUCTION**

#### **Creating a New Project**

- Create a new folder and name it "Hello"
- Copy the following files to the "Hello" folder
  - manaul code\Startup\system\_LPC17xx.c
  - manual\_code\UART\_polling\src\uart\_polling.c
  - manual\_code\UART\_polling\src\uart\_polling.h
- Create a new μVision by click
  - Project → New μVision Project



No Space allowed in the project folder path name on Nexus!

# **Choosing the Device**

Choose NXP(Founded by Philips) → LPC1768



### **Copying the startup Code**

Answer "Yes" to copy the startup code



- If you answer "No", you need to copy the file manually from the example code folder.
  - manual\_code\Startup\startup\_LPC17xx.s

### **Default Project Components**

You will see the default project setup as follows:



#### **Renaming Project Components**

- Rename the default target name
  - Click the target name to highlight it
  - Click the highlighted name to input a new target name, say "HelloWorld SIM"
- Rename the default source group name
  - Click the default source group to highlight it
  - Click the highlighted name to input a new name, say

"Startup Code"



# **Adding New Source Groups**

- Right click the target name and add the following source groups
  - System Code
  - Source Code
- You will see the following





### **Adding Files to Source Groups**

- Double click the "System Code" group and add the following files to it.
  - system\_LPC17xx.c



Double click file names to add or highlight file names and click "Add" button.

Add "uart\_polling.c" to "Source Code" Group

#### **Creating Your Own Source Code**

Your project would now look like this.



Click New button to Create a main.c file.



## Adding main.c to Project

• Add the main.c to the "Source Code" group.

```
д
Project
                          main.c*
1 #include <LPC17xx.h>
                          2 #include "uart polling.h"
  🚊 🚐 Startup Code
                          3 int main() {
     startup_LPC17xx.s
                               SystemInit();
  uart0 init();
     system_LPC17xx.c
                               uart0 put string("Hello World!\n\r");
  return 0:
     --- 🕍 uart_polling.c
     main.c
                          9
```

#### **Build and Download**

Click the Build button



Click the Download button



#### **RUN on the Board**



Press the "Reset" button on the board to run



# **Debug Configuration**





# **Debug by Simulator**



Debugger





# Debug by ULINK Cortex Debugger



#### **Example RAM.ini File**

```
FUNC void Setup (void) {
 SP = RDWORD(0x10000000); // Setup Stack Pointer
 PC = RDWORD(0x10000004); // Setup Program Counter
 // Setup Vector Table Offset Register
  WDWORD (0xE000ED08, 0x1000000);
LOAD Hello.axf INCREMENTAL // Download
                            // Setup for Running
Setup();
q, main
```



#### **RTX PROJECT P1 REQUIREMENTS**

#### P1 Requirements: API

 Memory Management: a memory pool which has fixed size of memory block and fixed number of memory blocks.

```
void *request_memory_block()
int release_memory_block(void *MemoryBlock)
```

Processor Management

```
int release_processor()
```

Process Priority Management

```
int set_process_priority(int process_ID, int priority)
int get_process_priority(int process_ID)
```

#### P1 Requirements: Processes

- Null Process
  - A system process which does nothing in an infinite loop. PID=0. All processes never terminate!
- Test Processes No new process created on the fly.
  - Up to six test processes with PIDs = 1,2, ..., 6
  - User level processes, only calls the user APIs
- Initialization
  - Memory, system processes and user processes

#### **MCB1700 HARDWARE**

#### Keil MCB 1700 Board

- Cortex-M3 Processor
- NXP LPC1768 MCU
- Up to 100 MHZ cpu
- One SystemTick Timer
- Four Timers
- Four UARTs
- Many other cool stuff...



#### **Cortex-M3 Overview**



(Image Courtesy of [1])

#### 32-bit microprocessor

- 32-bit data path
- 32-bit register bank
- 32-bit memory interface

#### Harvard Architecture

- Separate data and memory bus
- instruction and data buses share the same memory space (a unified memory system)

#### **Cortex-M3 Registers**

- General Purpose Registers (R0-R15)
  - Low registers (R0-R7)
    - 16-bit Thumb instructions and 32-bit Thumb-2 instructions
  - High registers (R8-R12)
    - All Thumb-2 instructions
  - Stack Pointer (R13)
    - MSP: Privileged, default after reset, os kernel, exception handler
    - **PSP:** Uer-level (i.e. unprivileged) base-level application
  - Link Register (R14)
  - Program Counter (R15)
- Special Registers
  - Program Status registers (PSRs)
  - Interrupt Mask registers (PRIMASK, FAULTMASK, and BASEPRI)
  - Control register (CONTROL)

#### Cortex-M3 Registers

32-bit microprocessor32-bit data path32-bit register bank32-bit memory interfaceHarvard ArchitectureSeparate data and memory bus

Low registers: R0-R7
16-bit Thumb instructions
32-bit Thumb-2 instructions
High registers: R9-R12

All Thumb-2 instructions

MSP: default after reset os kernel, exception handler Privileged PSP: base-level application

unprivileged, user-level



(Image Courtesy of [1])

# **Cortex-M3 Memory Map**

| 0xFFFF FFFF<br>0xE000 0000 | 0.5G | System level       | Private peripherals including NVIC, MPU and debug components     | $\Rightarrow$ |
|----------------------------|------|--------------------|------------------------------------------------------------------|---------------|
| 0xA000 0000                | 1.0G | External<br>device | Mainly used as external peripherals                              |               |
| 0x6000 0000                | 1.0G | External RAM       | Mainly used as external memory                                   |               |
| 0x4000 0000                | 0.5G | Peripherals        | Mainly used as peripherals                                       |               |
| 0x2000 0000                | 0.5G | SRAM               | Mainly used as static RAM                                        |               |
| 0x0000 0000                | 0.5G | Code               | Mainly used fro program code. Exception vector table after reset | *             |

(Table Courtesy of [1])

# **LPC1768 Memory Map**

| 0x2008 4000 |        |                              |  |
|-------------|--------|------------------------------|--|
|             |        |                              |  |
| 0x2007 C000 | 32 KB  | AHB SRAM (2 blocks of 16 KB) |  |
|             |        |                              |  |
| 0x1FFF 2000 |        | Reserved                     |  |
|             |        |                              |  |
| 0x1FFF 0000 | 8 KB   | Boot ROM                     |  |
|             |        |                              |  |
| 0x1000 8000 |        | Reserved                     |  |
|             |        |                              |  |
| 0x1000 0000 | 32 KB  | Local SRAM                   |  |
|             |        |                              |  |
| 0x0008 0000 |        | Reserved                     |  |
|             |        |                              |  |
| 0x0000 0000 | 512 KB | On-chip flash                |  |

# Example Flow Using ARM Development Tools



(Image Courtesy of [1])

#### Image memory layout

- A simple image consists of:
  - read-only (RO) section (Code + RO-data)
  - a read-write (RW) section (RW-data)
  - a zero-initialized (ZI) section (ZI-data)



### **End Address of the Image**

Linker defined symbol
 Image\$\$RW\_IRAM1\$\$ZI\$\$Limit

```
extern unsigned int Image$$RW_IRAM1$$ZI$$Limit;
unsigned int free_mem =
   (unsigned int) &Image$$RW_IRAM1$$ZI$$Limit;
```

# **Operation Modes**

- Two modes
  - Thread mode
  - Handler mode
- Two privilege levels
  - Privileged level
  - User level



(Image Courtesy of [1])

#### **OS Initialization Mode Switch**



### Exceptions (1)

- NVIC (Nested Vectored Interrupt Controller)
  - System Exceptions
    - Exception Numbers 1 -15
    - SVC call exception number is 11
  - External Interrupts
    - Exception Numbers 16-50
    - Timer0-3 IRQ numbers are 17-20
    - UARTO-3 IRQ numbers are 21-24
- Vector table is at 0x0 after reset.
- 32 programmable priorities
- Each vector table entry contains the exception handler's address (i.e. entry point)

## Exceptions (2)

| Address     | Exception Number | Value (Word Size)                            |
|-------------|------------------|----------------------------------------------|
| 0x0000 0000 | -                | MSP initial value                            |
| 0x0000 0004 | 1                | Reset vector (program counter initial value) |
| 0x0000 0008 | 2                | NMI handler starting address                 |
| 0x0000 000C | 3                | Hard fault handler starting address          |
| •••         |                  | Other handler starting address               |



### **Exception Stack Frame**



## CORTEX-M3 SOFTWARE DEVELOPMENT

# AAPCS (ARM Architecture Procedure Call Standard)

- R0-R3
  - Input parameters Px of a function. R0=P1, R1=P2, R2=P3 and R3=P4
  - R0 is used for return value of a function
- R12, SP, LR and PC
  - R12 is the Intra-Procedure-call scratch register.
- R4-R11
  - Must be preserved by the called function. C compiler generates push and pop assembly instructions to save and restore them automatically.

#### **CMSIS Structure**



#### **CMSIS Structure**

- Hardware Abstraction Layer (HAL) for Cortex-M processor registers
  - NVIC, MPU
- Standardized system exception names. For example:

```
void SVC_Handler();
void UARTO IRQHandler();
```

- Standardized method of header file organization
- Common method for system initialization

```
SystemInit()
```

Standardized intrinsic functions. For example:

```
void __disable_irq(void);
void enable irq(void);
```

- Common access functions for communication
- Standardized way for embedded software to determine system clock frequency
  - SystemFrequency variable is defined in device driver code

#### **CMSIS** Files



## **External Interrupt Programming**

- IRQn is defined in <device.h> file (i.e. LPC17xx.h)
- A set of functions
  - void NVIC EnableIRQ(IRQn Type IRQn)
  - void NVIC DisableIRQ(IRQn Type IRQn)
  - void NVIC\_SetPriority(IRQn\_Type IRQn, int32\_t priority)
  - uint32\_t NVIC\_GetPriority(IRQn\_Type IRQn)
  - void NVIC SetPendingIRQ(IRQn Type IRQn)
  - void NVIC\_ClearPendingIRQ(IRQn\_Type IRQn)
  - IRQn\_Type NVIC\_GetPendingIRQ(IRQn\_Type IRQn)

### **CMSIS Example**

```
#include "vendor_device.h" // For example.
  // lm3s_cmsis.h for LuminaryMicro devices
  // LPC17xx.h for NXP devices
                                                          Common name for
  // stm32f10x.h for ST devices
                                                       system initialization code
                                                    (from CMSIS v1.30, this function
void main(void)
                                                      is called from startup code)
  SystemInit():
                                                    NVIC setup by core access
  NVIC_SetPriority(UART1_IRQn. 0x0);
                                                            functions
  NVIC EnableIRQ(UART1 IRQn):
                                                    Interrupt numbers defined in
                                                        system <device>.h
void UART1_IRQHandler
                                                     Peripheral interrupt names are
                                                     device specific, define in device
                                                         specific startup code
void SysTick_Handler(void) (
                                                    System exception handler
       (Image Courtesy of [1])
                                                     names are common to all
                                                     Cortex microcontrollers
```

## **SVC** as a Gateway for OS Functions



## System calls through SVC in C

```
User Space int rls mem blk (void *)
                                               rtx.h
extern int k rls mem blk(void *);
#define SVC 0 svc indirect(0)
#define rls mem blk(blk)
       rls mem blk((U32)k rls mem blk, blk)
extern int rls mem blk (U32 p, void* blk) SVC 0
LDR.W r12, [pc, #offset]
                                       Generated by the compiler
        ;Load k rls mem blk in r12
SVC 0x00,
                                               HAL.C
SVC Handler: BLX R12
                                               rtx.c
Kernel Space int k rls mem blk(void*)
```

#### Hints

- Start with compile time memory pool and then change it later to dynamic memory pool.
- Start with just one ready queue and two simple user processes in your system to implement the context switching between the two processes.
- Once you get two processes working properly under context switching, add more ready queues to different priority levels and add user test process one by one to re-fine your context switching logic.

#### References

- 1. Yiu, Joseph, *The Definite Guide to the ARM Cortex-M3*, 2009
- 2. RealView® Compilation Tools Version 4.0 Developer Guide
- 3. ARM Software Development Toolkit Version 2.50 Reference Guide
- 4. LPC17xx User's Manual